Час 7 - понављање облика коришћењем петљи¶
Цртање у петљи¶
Концентрични кругови¶
Напиши програм који црта концентричне кругове чији је центар у центру екрана, а полупречници су редом 10, 20, …, 100 пиксела. Сваки круг нацртати црвеном линијом дебљином 5 пиксела.
Центар прозора се једноставно израчунава тако да му је координата x на пола ширине, а координата y на пола висине прозора (обрати пажњу на то да се код кругова морају користити цели бројеви, па стога морамо употребити целобројно дељење). Након тога, можемо навести 10 наредби за цртање кругова.
# bojimo pozadinu prozora u belu
prozor.fill(pg.Color("white"))
# centar kruga je u centru prozora
centar = (sirina // 2, visina // 2)
pg.draw.circle(prozor, pg.Color("red"), centar, 10, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 20, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 30, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 40, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 50, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 60, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 70, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 80, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 90, 5)
pg.draw.circle(prozor, pg.Color("red"), centar, 100, 5)
Претходно решење има пуно мана. За почетак, тешко је откуцати оволики број наредби. Даље, ако бисмо се предомислили и променили број кругова или величине полупречника, програм би захтевао доста компликоване измене. Задатак се много боље и једноставније решава када се примети да се полупречници кругова редом мењају од 10 до 100 са кораком 10 (10, 20, 30, …, 90, 100) и када се подсетимо да у Пајтону такво набрајање може остварити петљом:
for i in range(pocetak, kraj, korak):
...
Пошто десни крај није укључен у набрајање (узимају се вредности из полуотвореног интервала [pocetak,kraj)), да би последњи полупречник био 100, за крај је потребно навести вредност 101.
На основу претходне дискусије допуни наредни програм.
import pygame as pg
import pygamebg
(sirina, visina) = (300, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Концентрични кругови")
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
# centar kruga je u centru prozora - obrati pažnju na tip podataka
centar = (???, ???)
# poluprečnik se menja od 10 do 100, sa korakom 10
for r in range(10, ???, 10):
# crtamo krug
???
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Мердевине¶
Измените наредни програм тако да се пречаге мердевина цртају у петљи.
Пажљиво проучи које се вредности мењају кроз позиве функције
pg.draw.line
, шта је почетна вредност, шта је крајња вредност и
који је корак, па на основу тога позови функцију range
у оквиру
петље for
.
import pygame as pg
import pygamebg
(sirina, visina) = (300, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Мердевине")
prozor.fill(pg.Color("beige")) # bojimo pozadinu ekrana u bež
# leva strana
pg.draw.line(prozor, pg.Color("brown"), (100, 10), (100, visina - 10), 10)
# desna strana
pg.draw.line(prozor, pg.Color("brown"), (200, 10), (200, visina - 10), 10)
# ovaj deo prepraviti
pg.draw.line(prozor, pg.Color("brown"), (100, 50), (200, 50), 10) # prečaga
pg.draw.line(prozor, pg.Color("brown"), (100, 100), (200, 100), 10) # prečaga
pg.draw.line(prozor, pg.Color("brown"), (100, 150), (200, 150), 10) # prečaga
pg.draw.line(prozor, pg.Color("brown"), (100, 200), (200, 200), 10) # prečaga
pg.draw.line(prozor, pg.Color("brown"), (100, 250), (200, 250), 10) # prečaga
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Правилно распоређени бројеви¶
У оба претходна примера било је потребно да набројимо неки низ правилно распоређених бројева. У задатку са круговима то су били бројеви 10, 20, …, 100, а у задатку са мердевинама то су били бројеви 50, 100, 150, 200, 250.
У решењима задатка видели смо да је један начин да се то уради петља облика
for x in range(x0, xMax+1, dx):
...
при чему је било потребно обратити пажњу на то да десни крај није укључен у набрајање (узимају се вредности из полуотвореног интервала [x0,xMax+1)=[x0,xMax]).
Провери да ли ово разумеш тако што ћеш одговорити на следеће питање.
Наведимо још неколико начина да се исти ефекат постигне. Ако почетак обележимо са x0, а корак са dx, тада су вредности које исписујемо x0, x0+dx, x0+2dx, x0+3dx итд. Ако желимо да набројимо n ових вредности, тада можемо употребити петљу облика
for i in range(n):
x = x0 + i * dx
...
Још један начин је да променљиву x ажурирамо кроз сваки корак петље, тако што је увећавамо за dx.
x = x0
for i in range(n):
...
x += dx
Видећемо да се велики број задатака са цртањем правилно распоређених облика може решити применом оваквих петљи.
Нагласимо још и да функција range
са кораком (са три аргумента)
прима обавезно целобројне аргументе, па у ситуацијама када корак није
целобројан њено коришћење није могуће.
Хоризонтално и вертикално распоређивање облика¶
Често у применама имамо потребу да распоредимо објекте тако да буду један до другога, тако да су сви објекти равномерно распоређени, тј. тако да су свака два узастопна објекта на истом растојању.
Хоризонтално распоређени кругови¶
Нацртај 10 кругова пречника 30 пискела тако да буду равномерно распоређени ширином екрана и да се међусобно додирују.
Наредних пар питања ће ти помоћи да решиш овај задатак.
Q-49: Ако се два круга полупречника r додирују, тада је растојање између њихових центара једнако:
Ако круг полупречника r додирује леву ивицу екрана, тада је x координата његовог центра једнака:
На основу овога, допуни наредни програм.
import pygame as pg
import pygamebg
(sirina, visina) = (600, 100) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Хоризонтално распоређени кругови")
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
# crtamo 10 krugova
r = 30 # poluprečnik krugova
x = ??? # x koordinata centra kruga
for i in range(10):
# crtamo krug
pg.draw.circle(prozor, pg.Color("black"), (x, visina // 2), r, 1)
x += ??? # аžuriramo x tako da postane koordinata centra narednog kruga
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Још један начин да се овај задатак реши је тај да се примети да су координате центара кругова редом r, r+2r, r+4r, r+6r итд. Дакле, x координата круга са редним бројем i је r+2⋅i⋅r тј. (2i+1)r. Реши задатак коришћењем ове формуле.
import pygame as pg
import pygamebg
(sirina, visina) = (600, 100) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Хоризонтално распоређени кругови")
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
# crtamo 10 krugova
r = 30 # poluprečnik krugova
for i in range(10):
# crtamo krug
pg.draw.circle(prozor, pg.Color("black"), (???, visina // 2), r, 1)
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Трећи начин би могао искористити могућност да се функцијом range
може вршити набрајање са кораком (нпр. range(r, r + 10*2*r + 1,
2*r)
).
Вертикално распоређени кругови¶
Прикажимо сада како можемо распоредити кругове вертикално.
Напиши програм који црта кругове полупречника 10 пиксела равномерно распоређене вертикално средином екрана, тако да су им центри удаљени 50 пиксела (нацртај све кругове који се виде). Висина екрана се мења приликом сваког покретања програма.
Овај задатак је сличан претходном, уз неколико важних разлика. То што
су кругови распоређени вертикално уместо хоризонтално не мења пуно -
само је потребно заменити улогу x и y координата. Растојање између
полупречника је овај пут фиксно (износи 30 пиксела) и не израчунава се
на основу полупречника. Кључна разлика је то што број кругова није
унапред задат већ је кругове потребно цртати све док се бар неки њихов
делић види на екрану. Зато имамо две могућности. Или ћемо некако на
основу висине екрана израчунати број кругова који се виде или ћемо
уместо бројачке петље for
употребити условну петљу while
.
Ово друго може бити једноставније.
import random
import pygame as pg
import pygamebg
(sirina, visina) = (100, random.randint(150, 500)) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "")
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
r = 10 # poluprečnik krugova
dy = 30 # vertikalni razmak između centara dva uzastopna kruga
y = ??? # y koordinata centra tekućeg kruga
while ???:
pg.draw.circle(prozor, pg.Color("red"), (sirina // 2, y), r) # crtamo krug
y += ??? # centar narednog kruga je udaljen za dy od centra tekućeg kruga
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Правоугаона мрежа¶
Напиши програм који исцртава правоугаону мрежу која се састоји од 100 правоугаоних поља, распоређених у 10 врста и 10 колона (исцртати само линије мреже и то хоризонталне линије плавом бојом, а вертикалне црвеном, дебљине 5 пиксела).
Основни задатак је одредити координате x вертикалних линија и
координате y хоризонталних линија. Ширину једног правоугаоника можемо
одредити дељењем ширине прозора бројем колона (у нашем случају то је
10), док висину једног правоугаоника можемо одредити дељењем висине
прозора бројем врста (то је поново 10). Означимо те димензије са
dx и dy. Вертикалне линије се онда налазе на
растојању dx, 2dx, 3dx, …, 9dx пиксела од леве ивице прозора (то су им координате x). Пошто се
те линије простиру од врха до дна прозора, координате y су им једнаке
нули, односно висини прозора. Понављање цртања линија остварујемо,
наравно, употребом петље for
, при чему је најбоље да се бројач
i
креће од један до девет, јер се тада у кораку i
црта линија
од тачке (i*dx, 0)
до тачке (i*dx, visina)
. Цртање
хоризонталних линија остварујемо веома слично, у независној петљи
for
у којој се црта линија од тачке (0, i*dy)
до тачке
(sirina, i*dy)
.
import pygame as pg
import pygamebg
(sirina, visina) = (400, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Правоугаона мрежа")
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
brojPodeoka = 10
dx = sirina / brojPodeoka
dy = ??? # izračunaj razmak između podeoka po visini
# crtamo horizontalne linije
for i in range(1, brojPodeoka):
pg.draw.line(prozor, pg.Color("blue"), (0, i*dy), (sirina, i*dy), 5)
# dodaj kod koji crta vertikalne linije crvenom bojom
???
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Још један начин да се одреди координата наредне линије је да се координата претходне линије увећа за ширину тј. дужину правоугаоника.
x = dx
for i in range(1, brojPodeoka):
pg.draw.line(prozor, pg.Color("red"), (x, 0), (x, visina), 5)
x += dx
Шума¶
Уместо примитивних геометријских облика (кругова, правоугаоника, дужи и слично) можемо распоређивати сложеније облике и тако добити интересантне цртеже.
Нацртај шуму у којој су јелке распоређене једна до друге тако да се центар прве јелке налази на 100 пиксела од леве ивице екрана и да је размак између сваке две суседне јелке једнак 200 пиксела.
Јелку смо већ цртали у неком од претходних задатака. Претпоставићемо
зато да на располагању имамо функцију jelka(x, y)
која црта јелку
тако да јој је сидро постављено у тачку (x, y)
, при чему се сидро
налази на дну стабла, хоризонтално на његовој средини.
Допуни наредни кôд петљом у којој ће се позивати функција за цртање јелке.
import random
import pygame as pg
import pygamebg
(sirina, visina) = (500, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Шума")
def jelka(x, y):
# boje koje cemo koristiti
CRNA = (0, 0, 0)
ZELENA = (0, 100, 36)
BRAON = (97, 26, 9)
# stablo
pg.draw.rect(prozor, BRAON, (x-20, y-50, 40, 50))
# krošnja
pg.draw.polygon(prozor, ZELENA, [(x-100, y-50), (x+100, y-50), (x, y-150)])
pg.draw.polygon(prozor, ZELENA, [(x-75, y-100), (x+75, y-100), (x, y-200)])
pg.draw.polygon(prozor, ZELENA, [(x-50, y-150), (x+50, y-150), (x, y-250)])
# bojimo pozadinu prozora u belo
prozor.fill(pg.Color("white"))
???
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Штала¶
Још је лакше ако распоређујемо готове слике учитане из png
или
jpg
датотека.
У штали живи пет кравица. Напиши програм који их распоређује тако
да стоје једна поред друге, на дну прозора. Слику кравице учитај из
датотеке kravica.png
.
Пре него што решиш задатак, подсети се неколико основних функција у вези са сликама.
Пре слике са редним бројем i
налази се тачно i
слика, па се x
координата леве ивице те слике може добити множењем ширине слике са
i
. Пошто слика треба да буде на дну прозора, y координату њеног
горњег левог темена можемо израчунати тако што од висине прозора
одузмемо висину слике.
Допуни наредни програм на основу претходне дискусије.
import random
import pygame as pg
import pygamebg
(sirina, visina) = (750, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Кравице")
# bojimo pozadinu u belo
prozor.fill(pg.Color("white"))
# učitavamo sliku
slika = ???
# očitavamo dimenzije slike
(sirina_slike, visina_slike) = (slika.get_width(), slika.get_height())
# pet puta iscrtavamo sliku
for i in range(5):
prozor.blit(???)
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()
Наравно, могуће је било употребити и петљу наредног облика:
for x in range(0, sirina, sirina_slike):
prozor.blit(slika, (x, y))
Домаћи задатак - кућице¶
Напиши програм који исцртава низ белих кућица дуж једне улице. Kuћица има квадратну основу и кров у облику једнакостраничног троугла. Врата су постављена на средини основе, висина им је пола висине основе, а ширина четвртина ширине основе.
Дефинисаћемо прво функцију која црта једну кућицу. Цртање ћемо вршити релативно у односу на сидро постављено у доње лево теме кућице и димензију (представљену ширином кућице).
Уз помоћ претходног цртежа израчунај координате свих значајних тачака и допуни програм.
import math, random
import pygame as pg
import pygamebg
(sirina, visina) = (600, 300) # otvaramo prozor
prozor = pygamebg.open_window(sirina, visina, "Село")
# crta kućicu kojoj je donje levo teme u (x, y)
def kucica(x, y, dim_kucice):
(x0, y0) = (x, y - dim_kucice) # gornje levo teme osnove
# crtamo osnovu kuće
pg.draw.rect(prozor, pg.Color("white"), (x0, y0, dim_kucice, dim_kucice))
# crtamo krov (jednakostranični trougao)
krov_levo = (x0, y0) # donje levo teme krova
krov_desno = (x0 + dim_kucice, y0) # donje desno teme krova
krov_vrh = (???, ???) # vrh krova
pg.draw.polygon(prozor, pg.Color("red"), [???, ???, ???])
# uokvirujemo osnovu kuće
???
# crtamo vrata
sirina_vrata = dim_kucice / 4
visina_vrata = dim_kucice / 2
vrata_levo = ??? # x koordinata leve ivice vrata
vrata_gore = ??? # y koordinata gornje ivice vraa
pg.draw.rect(prozor, pg.Color("brown"), ???)
# bojimo pozadinu u nebo-plavo
prozor.fill(pg.Color("skyblue"))
# crtamo jednu po jednu kućicu
broj_kucica = 5
dim_kucice = sirina / broj_kucica
???
# prikazujemo prozor i čekamo da ga korisnik isključi
pygamebg.wait_loop()